专栏原创出处:github-源笔记文件 (opens new window) ,github-源码 (opens new window),欢迎 Star,转载请附上原文出处链接和本声明。
Scala 编程语言专栏系列笔记,系统性学习可访问个人复盘笔记-技术博客 Scala 编程语言 (opens new window)
# 什么是提取器对象
提取器对象是一个包含有 unapply 方法的单例对象。
unapply 方法接受一个实例对象然后返回最初创建它所用的参数。
提取器常用在模式匹配和偏函数中。
# unapply 返回结果定义
方式 1:提取单个 T 类型的值,
def unapply(obj: A): Option[T]
。方式 2:提取多个值,类型分别为 T1,...,Tn,可以把它们放在一个可选的元组中,
def unapply(obj: A): Option[(T1, ..., T2)]
。方式 3:判断真假,可以返回一个 Boolean 类型的值,
def unapply(obj: A): Boolean
。方式 4:提取值的数量不固定,根据输入来返回随机数量的值,提取方法必须定义为 unapplySeq,
def unapplySeq(obj: A): Option[Seq[T]]
。
# 方式 1-示例
提取单个值
import scala.util.Random object CustomerID { def apply(name: String) = s"$name--${Random.nextLong}" def unapply(customerID: String): Option[String] = { val stringArray: Array[String] = customerID.split("--") if (stringArray.tail.nonEmpty) Some(stringArray.head) else None } } // CustomerID("Sukyoung") 是简化语法,默认调用了 CustomerID.apply("Sukyoung"),返回一个字符串。 val customer1ID = CustomerID("Sukyoung") // Sukyoung--23098234908 customer1ID match { // 第一项匹配默认是在调用提取器方法 unapply,并将结果绑定到 name 上 case CustomerID(name) => println(name) // prints Sukyoung case _ => println("Could not extract a CustomerID") } val customer2ID = CustomerID("Nico") // 变量定义可以使用提取模式引入变量,提取器方法 unapply 可以用来初始化变量 name // 等价于 val name = CustomerID.unapply(customer2ID).get val CustomerID(name) = customer2ID println(name) // prints Nico
Copied!
# 方式 2-示例
提取多个值
trait School { def name: String def age: Int } class Student(val name: String, val age: Int, val score: Double) extends School class Teacher(val name: String, val age: Int) extends School object Student { def unapply(student: Student): Option[(String, Int, Double)] = Some((student.name, student.age, student.score)) } object Teacher { def unapply(teacher: Teacher): Option[(String, Int)] = Some((teacher.name, teacher.age)) } val lisa = new Student("Lisa", 14, 97) lisa match { // 不需要 age 属性,可以使用 _ 占位符代替 case Student(name, _, score) => println(s"学生 $name 获得 $score 分") case _ => println("老师没有评分资格") }
Copied!
# 方式 3-示例
提取 boolean 值
trait School { def name: String def age: Int } class Student(val name: String, val age: Int, val score: Double) extends School class Teacher(val name: String, val age: Int) extends School object Student { def unapply(student: Student): Option[(String, Int, Double)] = Some((student.name, student.age, student.score)) } object Teacher { def unapply(teacher: Teacher): Option[(String, Int)] = Some((teacher.name, teacher.age)) } object GoodStudent { def unapply(student: Student): Boolean = student.score > 90 } val lisa = new Student("Lisa", 14, 97) lisa match { // boolean 类型返回值模式匹配写法,student 相当于匹配到的对象实例 case student @ GoodStudent() => println(s"学生 ${student.name} 成绩优异") case _ => println("老师没有评分资格") }
Copied!
# 方式 4-示例
提取值数量不固定
object Split { def unapplySeq(str: String): Option[Seq[String]] = Some(str.split("-").toSeq) } val numberStr = "111-222-333-444" numberStr match { // 由于切分后得到的结果集合元素数量未知,因此使用该模式定义提取方法 first 为结果集第一个元素 case Split(first,tail @ _*) => println(s"第一个元素:$first ,剩余元素:$tail") case _ => println("Not match") }
Copied!
Gitalking ...